Estructuras dinámicas de datos
Consultas, lista de correo 'C++ Con Clase' 'C++ Con Clase' página de entrada Librerías estándar C Tabla de contenido Contactar con Webmaster
*Introducción
*1. Listas abiertas
*2. Pilas
*3. Colas
*4. Listas circulares
*5. Listas doblemente enlazadas
 . 5.1 Definición
 . 5.2 Tipos de datos
 . 5.3 Operaciones básicas
 . 5.4 Añadir elemento
 . 5.5 Buscar o localizar
 . 5.6 Eliminar elemento
 . 5.7 Ejemplo en C
 . 5.8 Ejemplo en C++
 . 5.9 Ejemplo C++ plantillas
*6. Árboles
*7. Árboles binarios de búsqueda (ABB)
*8. Árboles AVL
*Descarga de ejemplos
<< < > >>

5.7 Ejemplo de lista doblemente enlazada en C:  

Como en el caso de los ejemplos anteriores, construiremos una lista doblemente enlazada para almacenar números enteros. Para aprovechar mejor las posibilidades de estas listas, haremos que la lista esté ordenada. Haremos pruebas insertando varios valores, buscándolos y eliminándolos alternativamente para comprobar el resultado.

Algoritmo de inserción:

  1. El primer paso es crear un nodo para el dato que vamos a insertar.
  2. Si Lista está vacía, o el valor del primer elemento de la lista es mayor que el del nuevo, insertaremos el nuevo nodo en la primera posición de la lista.
  3. En caso contrario, buscaremos el lugar adecuado para la inserción, tenemos un puntero "anterior". Lo inicializamos con el valor de Lista, y avanzaremos mientras anterior->siguiente no sea NULL y el dato que contiene anterior->siguiente sea menor o igual que el dato que queremos insertar.
  4. Ahora ya tenemos anterior señalando al nodo adecuado, así que insertamos el nuevo nodo a continuación de él.
void Insertar(Lista *lista, int v) {
   pNodo nuevo, actual;

   /* Crear un nodo nuevo */
   nuevo = (pNodo)malloc(sizeof(tipoNodo));
   nuevo->valor = v;
   
   /* Colocamos actual en la primera posición de la lista */
   actual = *lista;
   if(actual) while(actual->anterior) actual = actual->anterior;
   
   /* Si la lista está vacía o el primer miembro es mayor que el nuevo */
   if(!actual || actual->valor > v) {
      /* Añadimos la lista a continuación del nuevo nodo */
      nuevo->siguiente = actual; 
      nuevo->anterior = NULL;
      if(actual) actual->anterior = nuevo;
      if(!*lista) *lista = nuevo;
   }
   else {
      /* Avanzamos hasta el último elemento o hasta que el siguiente tenga 
         un valor mayor que v */
      while(actual->siguiente &&actual->siguiente->valor <= v) 
         actual = actual->siguiente;
      /* Insertamos el nuevo nodo después del nodo anterior */
      nuevo->siguiente = actual->siguiente;
      actual->siguiente = nuevo;
      nuevo->anterior = actual;
      if(nuevo->siguiente) nuevo->siguiente->anterior = nuevo;
   }
}

Algoritmo de la función "Borrar":

  1. Localizamos el nodo de valor v
  2. ¿Existe?
    • SI:
      • ¿Es el nodo apuntado por lista?
        • SI: Hacer que lista apunte a otro sitio.
      • ¿Es el primer nodo de la lista?
        • NO: nodo->anterior->siguiente = nodo->siguiente
      • ¿Es el último nodo de la lista?
        • NO: nodo->siguiente->anterior = nodo->anterior
      • Borrar nodo
    void Borrar(Lista *lista, int v) {
       pNodo nodo;
       
       /* Buscar el nodo de valor v */
       nodo = *lista;
       while(nodo && nodo->valor <v) nodo = nodo->siguiente;
       while(nodo && nodo->valor > v) nodo = nodo->anterior;
    
       /* El valor v no está en la lista */
       if(!nodo || nodo->valor != v) return;
       
       /* Borrar el nodo */
       /* Si lista apunta al nodo que queremos borrar, apuntar a otro */
       if(nodo == *lista)
         if(nodo->anterior) *lista = nodo->anterior;
         else *lista = nodo->siguiente;
       
       if(nodo->anterior) /* no es el primer elemento */
          nodo->anterior->siguiente = nodo->siguiente;
       if(nodo->siguiente) /* no es el último nodo */
          nodo->siguiente->anterior = nodo->anterior;
       free(nodo);
    }

    Código del ejemplo completo:

    Tan sólo nos queda escribir una pequeña prueba para verificar el funcionamiento:

    #include <stdio.h>
    
    #define ASCENDENTE 1
    #define DESCENDENTE 0
    
    typedef struct _nodo {
       int valor;
       struct _nodo *siguiente;
       struct _nodo *anterior;
    } tipoNodo;
    
    typedef tipoNodo *pNodo;
    typedef tipoNodo *Lista;
    
    /* Funciones con listas: */
    void Insertar(Lista *l, int v);
    void Borrar(Lista *l, int v);
    
    void BorrarLista(Lista *);
    void MostrarLista(Lista l, int orden);
    
    int main() {
       Lista lista = NULL;
       pNodo p;
    
       Insertar(&lista, 20);
       Insertar(&lista, 10);
       Insertar(&lista, 40);
       Insertar(&lista, 30);
    
       MostrarLista(lista, ASCENDENTE);
       MostrarLista(lista, DESCENDENTE);
    
       Borrar(&lista, 10);
       Borrar(&lista, 15);
       Borrar(&lista, 45);
       Borrar(&lista, 30);
    
       MostrarLista(lista, ASCENDENTE);
       MostrarLista(lista, DESCENDENTE);
    
       BorrarLista(&lista);
    
       getchar();
       return 0;
    }
    
    void Insertar(Lista *lista, int v) {
       pNodo nuevo, actual;
    
       /* Crear un nodo nuevo */
       nuevo = (pNodo)malloc(sizeof(tipoNodo));
       nuevo->valor = v;
       
       /* Colocamos actual en la primera posición de la lista */
       actual = *lista;
       if(actual) while(actual->anterior) actual = actual->anterior;
       /* Si la lista está vacía o el primer miembro es mayor que el nuevo */
       if(!actual || actual->valor > v) {
          /* Añadimos la lista a continuación del nuevo nodo */
          nuevo->siguiente = actual; 
          nuevo->anterior = NULL;
          if(actual) actual->anterior = nuevo;
          if(!*lista) *lista = nuevo;
       }
       else {
          /* Avanzamos hasta el último elemento o hasta que el siguiente tenga 
             un valor mayor que v */
          while(actual->siguiente &&actual->siguiente->valor <= v) 
             actual = actual->siguiente;
          /* Insertamos el nuevo nodo después del nodo anterior */
          nuevo->siguiente = actual->siguiente;
          actual->siguiente = nuevo;
          nuevo->anterior = actual;
          if(nuevo->siguiente) nuevo->siguiente->anterior = nuevo;
       }
    }
    
    void Borrar(Lista *lista, int v) {
       pNodo nodo;
       
       /* Buscar el nodo de valor v */
       nodo = *lista;
       while(nodo && nodo->valor < v) nodo = nodo->siguiente;
       while(nodo && nodo->valor > v) nodo = nodo->anterior;
    
       /* El valor v no está en la lista */
       if(!nodo || nodo->valor != v) return;
       
       /* Borrar el nodo */
       /* Si lista apunta al nodo que queremos borrar, apuntar a otro */
       if(nodo == *lista)
         if(nodo->anterior) *lista = nodo->anterior;
         else *lista = nodo->siguiente;
       
       if(nodo->anterior) /* no es el primer elemento */
          nodo->anterior->siguiente = nodo->siguiente;
       if(nodo->siguiente) /* no es el último nodo */
          nodo->siguiente->anterior = nodo->anterior;
       free(nodo);
    }
    
    void BorrarLista(Lista *lista) {
       pNodo nodo, actual;
    
       actual = *lista;
       while(actual->anterior) actual = actual->anterior;
    
       while(actual) {
          nodo = actual;
          actual = actual->siguiente;
          free(nodo);
       }
       *lista = NULL;
    }
    
    void MostrarLista(Lista lista, int orden) {
       pNodo nodo = lista;
    
       if(!lista) printf("Lista vacía");
    
       nodo = lista;
       if(orden == ASCENDENTE) {
          while(nodo->anterior) nodo = nodo->anterior;
          printf("Orden ascendente: ");
          while(nodo) {
             printf("%d -> ", nodo->valor);
             nodo = nodo->siguiente;
          }
       }
       else {
          while(nodo->siguiente) nodo = nodo->siguiente;
          printf("Orden descendente: ");
          while(nodo) {
             printf("%d -> ", nodo->valor);
             nodo = nodo->anterior;
          }
       }
       
       printf("\n");
    }

    Fichero con el código fuente: Descargar programa

    << < > >>